<?php

/*  Copyright 2011  enlimbo lancers  (email : lancers@enlimbo.net)

  This program is free software; you can redistribute it and/or
  modify it under the terms of the GNU General Public License
  as published by the Free Software Foundation; either version 2
  of the License, or (at your option) any later version.

  This program is distributed in the hope that it will be useful,
  but WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  GNU General Public License for more details.

  You should have received a copy of the GNU General Public License
  along with this program; if not, write to the Free Software
  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 */

/**
 * Enlimbo Forms class for form creation
 *
 * @package Enlimbo
 * @subpackage Forms
 * @copyright enlimbo lancers 2012
 * @license GPLv2 or later
 * @version 1.1
 * @link http://enlimbo.net/forms
 * @author srdjan <srdjan@enlimbo.net>
 *
 *
 */
/*
  Element attributes

  General rules when adding attributes to an element using strings in chunks
 * always start the chunk with a space and do not end the chunk with another space
 * mind the closing tags and add a space when needed
 */

class Enlimbo_Forms {

	/**
	 * @var string
	 */
	private $_id;

	/**
	 * @var array
	 */
	private $_errors = array();

	/**
	 * @var array
	 */
	private $_elements = array();

	/**
	 * @var array
	 */
	private $_count = array();

	/**
	 * @var string
	 */
	public $css_class = 'wpt-form';

	/**
	 * @var string
	 */
	private $_validationFunc = '';

	/**
	 * @var array
	 */
	public $form_settings = array();

	public function __construct( $id ) {
		/**
		 * default settings
		 */
		$this->form_settings = array(
			'has_media_button' => true,
			'use_bootstrap' => false,
		);
		$this->_id = $id;
		if ( ! Toolset_Utils::is_real_admin() ) {
			// TODO check also doing_ajax as this gives false positives
			$cred_form_id = preg_replace( '/^cred_form_(\d+)_\d+$/', "$1", $this->_id );
			$form_settings = get_post_meta( $cred_form_id, '_cred_form_settings', true );
			if ( isset( $form_settings->form ) ) {
				$this->form_settings = $form_settings->form;
			}
			unset( $form_settings );
			/**
			 * check CRED setting for bootstrap: only on frontend
			 */
			$cred_cred_settings = get_option('cred_cred_settings');
			if (is_array($cred_cred_settings)) {
				$this->form_settings['use_bootstrap'] = array_key_exists('use_bootstrap', $cred_cred_settings) && $cred_cred_settings['use_bootstrap'];
			}
		}
	}

	/**
	 * Auto handler
	 *
	 * Renders.
	 *
	 * @param array $element
	 *
	 * @return HTML formatted output
	 */
	public function autoHandle( $id, $form ) {
		// Auto-add nonce field
		$form['nonce'] = array(
			'#type' => 'hidden',
			'#name' => '_nonce',
			'#value' => md5( $id ),
		);

		$this->_id = $id;
		$this->_elements = $form;

		// get submitted data
		if ( $this->isSubmitted() ) {

			// check if errors (validation)
			$this->validate( $this->_elements );

			// callback
			if ( empty( $this->_errors ) ) {

				if ( isset( $form['#form']['callback'] ) ) {
					if ( is_array( $form['#form']['callback'] ) ) {
						foreach ( $form['#form']['callback'] as $callback ) {
							if ( is_callable( $callback ) ) {
								call_user_func( $callback, $this );
							}
						}
					} else {
						if ( is_callable( $form['#form']['callback'] ) ) {
							call_user_func( $form['#form']['callback'], $this );
						}
					}
				}
				// Maybe triggered by callback function
				if ( empty( $this->_errors ) ) {
					// redirect
					if ( ! isset( $form['#form']['redirection'] ) ) {
						header( 'Location: ' . $_SERVER['REQUEST_URI'] );
					} elseif ( $form['#form']['redirection'] != false ) {
						header( 'Location: ' . $form['#form']['redirection'] );
					}
				}
			}
		}
	}

	/**
	 * Checks if form is submitted.
	 *
	 * @param type $id
	 *
	 * @return type
	 */
	public function isSubmitted( $id = '' ) {
		if ( empty( $id ) ) {
			$id = $this->_id;
		}

		return ( isset( $_REQUEST['_nonce'] ) && md5( $_REQUEST['_nonce'] ) == $id );
	}

	/**
	 * Sets validation function.
	 *
	 * @param type $class
	 */
	//    public function setValidationFunc($func)
	//    {
	//        $this->_validationFunc = $func;
	//    }

	/**
	 * Loops over elements and validates them.
	 *
	 * @param type $elements
	 */
	public function validate( &$elements ) {
		foreach ( $elements as $key => &$element ) {
			if ( ! isset( $element['#type'] ) || ! $this->_isValidType( $element['#type'] ) ) {
				continue;
			}
			if ( $element['#type'] != 'fieldset' ) {
				if ( isset( $element['#name'] ) && ! in_array( $element['#type'], array( 'submit', 'reset' ) ) ) {
					if ( $this->isSubmitted() ) {
						// Set submitted data
						if ( ! in_array( $element['#type'], array( 'checkboxes' ) ) && empty( $element['#forced_value'] ) ) {
							$element['#value'] = $this->getSubmittedData( $element );
						} elseif ( ! empty( $element['#options'] ) && empty( $element['#forced_value'] ) ) {
							foreach ( $element['#options'] as $option_key => $option ) {
								$option['#type'] = 'checkbox';
								$element['#options'][ $option_key ]['#value'] = $this->getSubmittedData( $option );
							}
						}
					}
				}
				// Validate
				if ( isset( $element['#validate'] ) ) {
					$this->validateElement( $element );
				}
			} elseif ( isset( $element['#type'] ) && $element['#type'] == 'fieldset' ) {
				$this->validate( $element );
			} elseif ( is_array( $element ) ) {
				$this->validate( $element );
			}
		}
	}

	/**
	 * Validates element.
	 *
	 * @param type $element
	 */
	public function validateElement( &$element ) {
		$value = isset( $element['#value'] ) ? $element['#value'] : null;
		if ( is_null( $value ) && isset( $element['#default_value'] ) ) {
			$value = $element['#default_value'];
		}
		$element = apply_filters( 'wptoolset_form_' . $this->_id . '_validate_field', $element, $value );
		if ( isset( $element['error'] ) ) {
			$this->_errors = true;
			$_errors = $element['error']->get_error_data();
			$element['#error'] = $_errors[0];
		}
	}

	/**
	 * Checks if there are errors.
	 *
	 * @return type
	 */
	public function isError() {
		return $this->_errors;
	}

	/**
	 * Sets errors to true.
	 */
	public function triggerError() {
		$this->_errors = true;
	}

	/**
	 * Renders form.
	 *
	 * @return type
	 */
	public function renderForm() {
		// loop over elements and render them
		return $this->renderElements( $this->_elements );
	}

	/**
	 * Counts element types.
	 *
	 * @param type $type
	 *
	 * @return type
	 */
	private function _count( $type ) {
		if ( ! isset( $this->_count[ $type ] ) ) {
			$this->_count[ $type ] = 0;
		}
		$this->_count[ $type ] += 1;

		return $this->_count[ $type ];
	}

	/**
	 * Check if element is of valid type
	 *
	 * @param string $type
	 *
	 * @return boolean
	 */
	private function _isValidType( $type ) {
		return in_array( $type, array(
			'select',
			'checkboxes',
			'checkbox',
			'radios',
			'radio',
			'textfield',
			'textarea',
			'file',
			'submit',
			'reset',
			'hidden',
			'fieldset',
			'markup',
			'button',
			'password',
		) );
	}

	/**
	 * Renders elements.
	 *
	 * @param type $elements
	 *
	 * @return type
	 */
	public function renderElements( $elements ) {
		$output = '';
		if ( ! isset( $elements ) ) {
			return $output;
		}
		foreach ( $elements as $key => $element ) {
			if ( ! isset( $element['#type'] ) || ! $this->_isValidType( $element['#type'] ) ) {
				continue;
			}
			if ( $element['#type'] != 'fieldset' ) {

				/**
				 * Temporary fixing validation for checkbox/radios/skype because _cakeValidation is not working for thats
				 * https://icanlocalize.basecamphq.com/projects/7393061-toolset/todo_items/186243370/comments
				 */
				if ( ! Toolset_Utils::is_real_admin() ) {
					if ( $element['#type'] == 'radios' ) {
						if ( isset( $element['#error'] ) ) {
							if ( isset( $element['#options'] ) && count( $element['#options'] ) > 0 && ! isset( $element['#options'][0]['#error'] ) ) {
								$element['#options'][0]['#error'] = $element['#error'];
							}
						}
					}
				}
				//##################################################################################################

				$output .= $this->renderElement( $element );
			} elseif ( isset( $element['#type'] ) && $element['#type'] == 'fieldset' ) {
				$buffer = $this->renderElements( $element );
				$output .= $this->fieldset( $element, 'wrap', $buffer );
			} elseif ( is_array( $element ) ) {
				$output .= $this->renderElements( $element );
			}
		}

		return $output;
	}

	/**
	 * Renders element.
	 *
	 * Depending on element type, it calls class methods.
	 *
	 * @param array $element
	 *
	 * @return HTML formatted output
	 */
	public function renderElement( $element ) {
		$method = $element['#type'];
		if ( ! isset( $element['#name'] ) && ! in_array( $element['#type'], array( 'markup', 'checkboxes' ) ) ) {
			if ( ! isset( $element['#attributes']['name'] ) ) {
				return '#name or #attributes[\'name\'] required!';
			} else {
				$element['#name'] = $element['#attributes']['name'];
			}
		}
		if ( is_callable( array( $this, $method ) ) ) {
			$custom_field_title = '';
			if ( isset( $element['#title'] ) && ! empty( $element['#title'] ) ) {
				$custom_field_title = $element['#title'];
			}

			if ( empty( $custom_field_title ) && isset( $element['#name'] ) && ! empty( $element['#name'] ) ) {
				$custom_field_title = $element['#name'];
			}
			if ( ! isset( $element['#id'] ) ) {
				if ( isset( $element['#attributes']['id'] ) ) {
					$element['#id'] = $element['#attributes']['id'];
				} else {
					$_id = isset( $this->_id ) ? $this->_id . '-' : '';
					$element['#id'] = "{$_id}{$element['#type']}-"
						. $this->_count( $element['#type'] ) . '-' . time();
				}
			}

			if ( isset( $this->_errors[ $element['#id'] ] ) ) {
				$element['#error'] = $this->_errors[ $element['#id'] ];
			}
			// Add JS validation
			if ( ! empty( $element['#validate'] ) ) {
				if ( isset( $element['#validate']['required'] ) && ! empty( $element['#title'] ) ) {
					// Asterisk
					$element['#title'] .= '&#42;';
				}
				$element['#attributes']['data-wpt-validate'] = esc_html( self::json_encode( apply_filters( 'wptoolset_forms_field_js_validation_data_' . $this->_id, $element['#validate'] ) ) );
				$element['#attributes']['data-wpt-field-title'] = esc_js( $custom_field_title );
			}
			if ( $element['#type'] == 'radios' && ! empty( $element['#options'] ) ) {
				foreach ( $element['#options'] as &$option ) {
					if ( ! empty( $option['#validate'] ) ) {
						$option['#attributes']['data-wpt-validate'] = esc_html( self::json_encode( apply_filters( 'wptoolset_forms_field_js_validation_data_' . $this->_id, $option['#validate'] ) ) );
						$option['#attributes']['data-wpt-field-title'] = esc_js( $custom_field_title );
					}
				}
			}
			/**
			 * WPML - lock CF is has option "copy from original".
			 */
			if ( Toolset_Utils::is_real_admin() && function_exists( 'wpcf_wpml_field_is_copied' ) && wpcf_wpml_field_is_copied( $element ) ) {
				$element['#title'] .= sprintf(
					'<img src="%s/images/locked.png" alt="%s" title="%s" style="position:relative;left:2px;top:2px;" />', WPCF_EMBEDDED_RES_RELPATH, __( 'This field is locked for editing because WPML will copy its value from the original language.', 'wpcf' ), __( 'This field is locked for editing because WPML will copy its value from the original language.', 'wpcf' )
				);
				$element['#attributes']['readonly'] = true;
				$element['#attributes']['disabled'] = true;
			}

			return $this->{$method}( $element );
		}
	}

	/**
	 * Sets other element attributes.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	private function _setElementAttributes( $element ) {
		$attributes = '';

		$classes = array();
		$classes[] = $this->css_class . '-' . $element['#type'];
		$classes[] = 'form-' . $element['#type'];
		
		if ($this->form_settings['use_bootstrap']) {
			switch ($element['#type']) {
				case 'hidden':
				case 'button':
				case 'submit':
				case 'radio':
				case 'checkbox':
				case 'file':
					//cred-162
				case 'option':
					break;
				default:
					$classes[] = 'form-control';
			}
		} else {
			if ('hidden' != $element['#type']) {
				$classes[] = $element['#type'];
			}
		}

		if ( 'hidden' != $element['#type'] ) {
			$classes[] = $element['#type'];
		}

		if ( isset( $element['#attributes'] ) && ! empty( $element['#attributes'] )
		) {
			if ( isset( $element['#attributes']['class'] ) ) {
				$element['#attributes']['class'] .= ' ' . implode( ' ', $classes );
			} else {
				$element['#attributes']['class'] = implode( ' ', $classes );
			}
		} else {
			$element['#attributes'] = array(
				'class' => implode( ' ', $classes ),
			);
		}

		foreach ( $element['#attributes'] as $attribute => $value ) {
			// Prevent undesired elements
			if ( in_array( $attribute, array( 'id', 'name', 'options', 'actual_value', 'actual_options' ) ) ) {
				continue;
			}
			// Don't set disabled for checkbox
			if (
				$element['#type'] == 'checkbox' && (
					$attribute == 'disabled' || $attribute == 'readonly'
				)
			) {
				continue;
			}

			// Set return string
			$attributes .= ' ' . $attribute . '="' . $value . '"';
		}

		return $attributes;
	}

	/**
	 * Sets render elements.
	 *
	 * @param array $element
	 */
	private function _setRender( $element ) {
		if ( ! isset( $element['#id'] ) ) {
			if ( isset( $element['#attributes']['id'] ) ) {
				$element['#id'] = $element['#attributes']['id'];
			} else {
				$element['#id'] = 'form-' . mt_rand();
			}
		}
		$element['_attributes_string'] = $this->_setElementAttributes( $element );
		$element['_render'] = array();
		$element['_render']['prefix'] = isset( $element['#prefix'] ) ? $element['#prefix'] . "\r\n" : '';
		$element['_render']['suffix'] = isset( $element['#suffix'] ) ? $element['#suffix'] . "\r\n" : '';
		$element['_render']['before'] = isset( $element['#before'] ) ? $element['#before'] . "\r\n" : '';
		$element['_render']['after'] = isset( $element['#after'] ) ? $element['#after'] . "\r\n" : '';
		$element['_render']['title'] = $this->_setElementTitle( $element );
		$element['_render']['description'] = isset( $element['#description'] ) ? $this->_setElementDescription( $element ) : '';
		$element['_render']['error'] = $this->renderError( $element ) . "\r\n";
		/**
		 * label
		 */
		$element['_render']['label'] = '';
		if ( isset( $element['#title'] ) ) {
			$classes = array();
			$classes[] = sprintf( '%s-label', $this->css_class );
			$classes[] = sprintf( '%s-%s-label', $this->css_class, $element['#type'] );

			$element['_render']['label'] .= sprintf(
				'<label class="%s" for="%s">', implode( ' ', $classes ), $element['#id']
			);
			$element['_render']['label'] .= stripslashes( $element['#title'] );
			$element['_render']['label'] .= '</label>';
		}

		return $element;
	}

	/**
	 * Applies pattern to output.
	 *
	 * Pass element property #pattern to get custom renedered element.
	 *
	 * @param array $pattern
	 *      Accepts: <prefix><suffix><label><title><desription><error>
	 * @param array $element
	 */
	private function _pattern( $pattern, $element ) {
		$pattern = strtolower( $pattern );
		foreach ( $element['_render'] as $key => $value ) {
			$pattern = str_replace( '<' . strtolower( $key ) . '>', $value, $pattern );
		}

		return $pattern;
	}

	/**
	 * Wrapps element in <div></div>.
	 *
	 * @param arrat $element
	 * @param string $output
	 *
	 * @return string
	 */
	private function _wrapElement( $element, $output ) {
		if ( ! empty( $element['#inline'] ) ) {
			return $output;
		}
		$classes = array();
		$classes[] = 'form-item';
		$classes[] = 'form-item-' . $element['#type'];
		$classes[] = $this->css_class . '-item';
		$classes[] = $this->css_class . '-item-' . $element['#type'];
		if ($this->form_settings['use_bootstrap']) {
			$classes[] = 'form-group';
		}
		if ( preg_match( '/_hidden$/', $element['#id'] ) && ! is_admin() ) {
			$classes[] = 'wpt-form-hide-container';
		}
		if ( Toolset_Utils::is_real_admin() ) {
			return sprintf(
				'<div id="%s-wrapper" class="%s">%s</div>', $element['#id'], implode( ' ', $classes ), $output
			);
		}

		return $output;
	}

	/**
	 * Returns HTML formatted output for element's title.
	 *
	 * @param string $element
	 *
	 * @return string
	 */
	private function _setElementTitle( $element ) {
		$output = '';
		if ( isset( $element['#title'] ) ) {
			$output .= '<div class="title '
				. $this->css_class . '-title '
				. $this->css_class . '-title-' . $element['#type'] . ' '
				. 'title-' . $element['#type'] . '">'
				. stripslashes( $element['#title'] )
				. "</div>\r\n";
		}

		return $output;
	}

	/**
	 * Returns HTML formatted output for element's description.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	private function _setElementDescription( $element ) {
		if ( empty( $element['#description'] ) ) {
			return '';
		}
		$element['#description'] = stripslashes( $element['#description'] );
		$output = "\r\n"
			. '<div class="description '
			. $this->css_class . '-description '
			. $this->css_class . '-description-' . $element['#type'] . ' '
			. 'description-' . $element['#type'] . '">'
			. $element['#description'] . "</div>\r\n";

		return $output;
	}

	/**
	 * Returns HTML formatted element's error message.
	 *
	 * Pass #supress_errors in #form element to avoid error rendering.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function renderError( $element ) {
		if ( ! isset( $element['#error'] ) ) {
			return '';
		}
		$output = '<label class="' . $this->css_class . '-error" for="'
			. $element['#id'] . '" generated="true">'
			. $element['#error'] . '</label>' . "\r\n";
		//        $output = '<div class="form-error '
		//                . $this->css_class . '-error '
		//                . $this->css_class . '-form-error '
		//                . $this->css_class . '-' . $element['#type'] . '-error '
		//                . $element['#type'] . '-error form-error-label'
		//                . '">' . $element['#error'] . '</div>'
		//                . "\r\n";
		return $output;
	}

	/**
	 * Returns HTML formatted output for fieldset.
	 *
	 * @param array $element
	 * @param string $action open|close|wrap
	 * @param string $wrap_content HTML formatted output of child elements
	 *
	 * @return string
	 */
	public function fieldset( $element, $action = 'open', $wrap_content = '' ) {
		$collapsible_open = '<div class="fieldset-wrapper">';
		$collapsible_close = '</div>';
		$legend_class = '';
		if ( ! isset( $element['#id'] ) ) {
			$element['#id'] = 'fieldset-' . $this->_count( 'fieldset' );
		}
		if ( ! isset( $element['_attributes_string'] ) ) {
			$element['_attributes_string'] = $this->_setElementAttributes( $element );
		}
		if ( ( isset( $element['#collapsible'] ) && $element['#collapsible'] ) || ( isset( $element['#collapsed'] ) && $element['#collapsed'] ) ) {
			$collapsible_open = '<div class="collapsible fieldset-wrapper">';
			$collapsible_close = '</div>';
			$legend_class = ' class="legend-expanded"';
		}
		if ( isset( $element['#collapsed'] ) && $element['#collapsed'] ) {
			$collapsible_open = str_replace( 'class="', 'class="collapsed ', $collapsible_open );
			$legend_class = ' class="legend-collapsed"';
		}
		$output = '';
		switch ( $action ) {
			case 'close':
				$output .= $collapsible_close . "</fieldset>\r\n";
				$output .= isset( $element['#suffix'] ) ? $element['#suffix']
					. "\r\n" : '';
				$output .= "\n\r";
				break;

			case 'open':
				$output .= $collapsible_open;
				$output .= isset( $element['#prefix'] ) ? $element['#prefix']
					. "\r\n" : '';
				$output .= '<fieldset' . $element['_attributes_string']
					. ' id="' . $element['#id'] . '">' . "\r\n";
				$output .= isset( $element['#title'] ) ? '<legend'
					. $legend_class . '>'
					. stripslashes( $element['#title'] )
					. "</legend>\r\n" : '';
				$output .= isset( $element['#description'] ) ? $this->_setElementDescription( $element ) : '';
				$output .= "\n\r";
				break;

			case 'wrap':
				if ( ! empty( $wrap_content ) ) {
					$output .= isset( $element['#prefix'] ) ? $element['#prefix'] : '';
					$output .= '<fieldset' . $element['_attributes_string']
						. ' id="' . $element['#id'] . '">' . "\r\n";
					$output .= '<legend' . $legend_class . '>'
						. stripslashes( $element['#title'] )
						. "</legend>\r\n"
						. $collapsible_open;
					$output .= isset( $element['#description'] ) ? $this->_setElementDescription( $element ) : '';
					$output .= $wrap_content . $collapsible_close
						. "</fieldset>\r\n";
					$output .= isset( $element['#suffix'] ) ? $element['#suffix'] : '';
					$output .= "\n\r";
				}
				break;
		}

		return $output;
	}

	/**
	 * Returns HTML formatted output for checkbox element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function checkbox( $element ) {
		$element['#type'] = 'checkbox';
		$element = $this->_setRender( $element );
		$element['_render']['element'] = '<input type="checkbox"';
		foreach ( array( 'id', 'name' ) as $key ) {
			$element['_render']['element'] .= sprintf( ' %s="%s"', $key, $element[ '#' . $key ] );
		}
		/**
		 * type and data_id
		 */
		$element['_render']['element'] .= sprintf( ' data-wpt-type="%s"', __FUNCTION__ );
		$element['_render']['element'] .= $this->_getDataWptId( $element );

		/*
		 * Add html attribute value=""
		 *
		 * We have key #value and we also have #default-value key
		 * #value
		 *   CRED:  fills this with what value="" of input should be
		 *   TYPES: fills this with the stored database value (WRONG)
		 *
		 * #default-value (this should better be called #checked)
		 *   CRED:  uses this to determine if the checkbox is checked or not (empty / 1)
		 *   TYPES: uses this for the "value to store" input when creating a checkbox field (WRONG)
		 *
		 * To get things right we flip the usage of Types fields to handle it the same way as CRED does
		 *
		 * The fact that a field comes from Types is determined from its name (sic!). This is very fragile.
		 *
		 * START Todo: Types should deliver the correct values instead of flipping it here
		 */
		$is_types_field = (
			( strpos( $element['#name'], 'wpcf[' ) === 0 ) || ( strpos( $element['#name'], 'wpcf_post_relationship[' ) === 0 )
		);

		if ( $is_types_field ) {
			$tmp_value = $element['#value'];

			$element['#value'] = array_key_exists( '#default_value', $element ) ? $element['#default_value'] : '';

			$element['#default_value'] = $tmp_value;
		}
		/* END To-do */


		$element['_render']['element'] .= ' value="';

		// Specific: if value is empty force 1 to be rendered
		// but if is defined default value, use default
		$value = 1;
		if ( array_key_exists( '#default_value', $element ) ) {
			$value = $element['#default_value'];
		}

		// Decide whether we should use the default value from field definition
		// or the current value this checkbox has (from database, presumably)
		//
		// fixme this logic needs to be reviewed and simplified
		$is_zero = ! preg_match( '/^0$/', $element['#value'] );
		$is_empty = empty( $element['#value'] );
		$is_boolean = is_bool( $element['#value'] );
		$use_default_value = ( ( $is_empty && $is_zero ) || $is_boolean );
		$value_output = ( $use_default_value ? $value : esc_attr( $element['#value'] ) );
		$element['_render']['element'] .= $value_output;

		$element['_render']['element'] .= '"' . $element['_attributes_string'];
		if (
			(
				! $this->isSubmitted() && (
					( ! empty( $element['#default_value'] ) && $element['#default_value'] == $element['#value'] ) || ( isset( $element['#checked'] ) && $element['#checked'] )
				)
			) || ( $this->isSubmitted() && ! empty( $element['#value'] ) )
		) {
			$element['_render']['element'] .= ' checked="checked"';
		}
		if ( ! empty( $element['#attributes']['disabled'] ) || ! empty( $element['#disable'] ) ) {
			$element['_render']['element'] .= ' onclick="javascript:return false; if(this.checked == 1){this.checked=1; return true;}else{this.checked=0; return false;}"';
		}
		$element['_render']['element'] .= ' />';

		$pattern = $this->_getStatndardPatern( $element, '<BEFORE><PREFIX><ELEMENT>&nbsp;<LABEL><ERROR><SUFFIX><DESCRIPTION><AFTER>' );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output . "\r\n";
	}

	/**
	 * Returns HTML formatted output for checkboxes element.
	 *
	 * Renders more than one checkboxes provided as elements in '#options'
	 * array element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function checkboxes( $element ) {
		$element['#type'] = 'checkboxes';
		$element = $this->_setRender( $element );
		$clone = $element;
		$clone['#type'] = 'checkbox';
		$element['_render']['element'] = '';
		$allowed_inherited_attributes = array( 'disabled' );
		foreach ( $element['#options'] as $ID => $value ) {
			if ( ! is_array( $value ) ) {
				$value = array( '#title' => $ID, '#value' => $value, '#name' => $element['#name'] . '[]' );
			}
			foreach ( $allowed_inherited_attributes as $allowed_in_att ) {
				if ( isset( $element['#attributes'][ $allowed_in_att ] ) ) {
					$value['#attributes'][ $allowed_in_att ] = $element['#attributes'][ $allowed_in_att ];
				}
			}
			$element['_render']['element'] .= $this->checkbox( $value );
		}
		$pattern = $this->_getStatndardPatern( $element, '<BEFORE><PREFIX><TITLE><DESCRIPTION><ELEMENT><SUFFIX><AFTER>' );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output;
	}

	/**
	 * Returns HTML formatted output for radio element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function radio( $element ) {
		$element['#type'] = 'radio';
		$element = $this->_setRender( $element );
		$element['_render']['element'] = '<input type="radio" id="'
			. $element['#id'] . '" name="'
			. $element['#name'] . '" value="';
		$element['_render']['element'] .= isset( $element['#value'] ) ? htmlspecialchars( $element['#value'] ) : $this->_count['radio'];
		$element['_render']['element'] .= '"';
		$element['_render']['element'] .= $element['_attributes_string'];
		$element['_render']['element'] .= ( isset( $element['#value'] ) && $element['#value'] === $element['#default_value'] ) ? ' checked="checked"' : '';
		if ( isset( $element['#disable'] ) && $element['#disable'] ) {
			$element['_render']['element'] .= ' disabled="disabled"';
		}
		if ( array_key_exists( '#types-value', $element ) ) {
			$element['_render']['element'] .= sprintf( ' data-types-value="%s"', $element['#types-value'] );
		}
		/**
		 * type and data_id
		 */
		$element['_render']['element'] .= sprintf( ' data-wpt-type="%s"', __FUNCTION__ );
		$element['_render']['element'] .= $this->_getDataWptId( $element );

		$element['_render']['element'] .= ' />';

		$pattern = isset( $element['#pattern'] ) ? $element['#pattern'] : '<BEFORE><PREFIX><ELEMENT>&nbsp;<LABEL><ERROR><SUFFIX><DESCRIPTION><AFTER>';
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output . "\r\n";
	}

	/**
	 * Returns HTML formatted output for radios elements.
	 *
	 * Radios are provided via #options array.
	 * Requires #name value.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function radios( $element ) {
		if ( ! isset( $element['#name'] ) || empty( $element['#name'] ) ) {
			return false;
		}
		$element['#type'] = 'radios';
		$element = $this->_setRender( $element );
		$element['_render']['element'] = '';
		foreach ( $element['#options'] as $ID => $value ) {
			$this->_count( 'radio' );
			if ( ! is_array( $value ) ) {
				$value = array( '#title' => $ID, '#value' => $value );
				$value['#inline'] = true;
				$value['#after'] = '<br />';
			}
			$value['#name'] = $element['#name'];
			$value['#default_value'] = isset( $element['#default_value'] ) ? $element['#default_value'] : $value['#value'];
			$value['#disable'] = isset( $element['#disable'] ) ? $element['#disable'] : false;
			if (
				isset( $element['#attributes']['disabled'] ) && $element['#attributes']['disabled']
			) {
				$value['#disable'] = true;
			}
			$element['_render']['element'] .= $this->radio( $value );
		}
		if ( Toolset_Utils::is_real_admin() ) {
			$pattern = '<BEFORE><PREFIX><TITLE><DESCRIPTION><ELEMENT><ERROR><SUFFIX><AFTER>';
		} else {
			$pattern = '<BEFORE><PREFIX><DESCRIPTION><ELEMENT><ERROR><SUFFIX><AFTER>';
		}

		$pattern = $this->_getStatndardPatern( $element, $pattern );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output;
	}

	/**
	 * Returns HTML formatted output for select element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function select( $element ) {
		$element = $this->_setRender( $element );

		$element['_render']['element'] = '';
		$element['_render']['element'] .= '<select id="' . $element['#id'] . '" ';
		$element['_render']['element'] .= $element['_attributes_string'];
		$element['_render']['element'] .= sprintf( ' data-wpt-type="%s"', __FUNCTION__ );
		/**
		 * multiple
		 */
		if ( array_key_exists( '#multiple', $element ) && $element['#multiple'] ) {
			$element['_render']['element'] .= ' multiple="multiple"';
			$element['_render']['element'] .= ' name="' . $element['#name'] . '[]"';
		} else {
			$element['_render']['element'] .= ' name="' . $element['#name'] . '"';
		}
		$element['_render']['element'] .= ">\r\n";
		$count = 1;
		foreach ( $element['#options'] as $id => $value ) {
			if ( ! is_array( $value ) ) {
				$value = array( '#title' => $id, '#value' => $value, '#type' => 'option' );
			}
			$value['#type'] = 'option';
			if ( ! isset( $value['#value'] ) ) {
				$value['#value'] = $this->_count['select'] . '-' . $count;
				$count += 1;
			}
			$element['_render']['element'] .= '<option value="' . htmlspecialchars( $value['#value'] ) . '"';
			$element['_render']['element'] .= $this->_setElementAttributes( $value );
			if ( array_key_exists( '#types-value', $value ) ) {
				$element['_render']['element'] .= sprintf( ' data-types-value="%s"', $value['#types-value'] );
			}
			/**
			 * type and data_id
			 */
			$element['_render']['element'] .= ' data-wpt-type="option"';
			$element['_render']['element'] .= $this->_getDataWptId( $element );
			/**
			 * selected
			 */
			if ( array_key_exists( '#multiple', $element ) && $element['#multiple'] ) {
				if ( is_array( $element['#default_value'] ) && in_array( $value['#value'], $element['#default_value'] ) ) {
					$element['_render']['element'] .= ' selected="selected"';
				}
			} elseif ( $element['#default_value'] == $value['#value'] ) {
				$element['_render']['element'] .= ' selected="selected"';
			}
			$element['_render']['element'] .= '>';
			$element['_render']['element'] .= isset( $value['#title'] ) ? $value['#title'] : $value['#value'];
			$element['_render']['element'] .= "</option>\r\n";
		}
		$element['_render']['element'] .= '</select>';
		$element['_render']['element'] .= PHP_EOL;

		$pattern = $this->_getStatndardPatern( $element );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output;
	}

	/**
	 * Returns HTML formatted output for textfield element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function textfield( $element ) {
		$element['#type'] = 'textfield';
		$element = $this->_setRender( $element );

		$element['_render']['element'] = '<input type="text"';
		//$element['_render']['element'] .= sprintf( ' data-wpt-type="%s" ', __FUNCTION__ );
		$element['_render']['element'] .= sprintf( ' id="%s"', $element['#id'] );
		$element['_render']['element'] .= sprintf( ' name="%s"', $element['#name'] );

		$value = toolset_getarr( $element, '#value' );
		$value = ( is_string( $value ) ? esc_attr( $value ) : '' );
		$element['_render']['element'] .= sprintf( ' value="%s"', $value );

		$element['_render']['element'] .= $element['_attributes_string'];
		if ( isset( $element['#disable'] ) && $element['#disable'] ) {
			$element['_render']['element'] .= ' disabled="disabled"';
		}
		/**
		 * type and data_id
		 */
		$element['_render']['element'] .= sprintf( ' data-wpt-type="%s"', __FUNCTION__ );
		$element['_render']['element'] .= $this->_getDataWptId( $element );

		$element['_render']['element'] .= ' />';
		$pattern = $this->_getStatndardPatern( $element );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output . "\r\n";
	}

	/**
	 * Returns HTML formatted output for textfield element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function password( $element ) {
		$element['#type'] = 'password';
		$element = $this->_setRender( $element );
		$element['_render']['element'] = '<input type="password" id="'
			. $element['#id'] . '" name="' . $element['#name'] . '" value="';
		$element['_render']['element'] .= isset( $element['#value'] ) ? $element['#value'] : '';
		$element['_render']['element'] .= '"' . $element['_attributes_string'];
		if ( isset( $element['#disable'] ) && $element['#disable'] ) {
			$element['_render']['element'] .= ' disabled="disabled"';
		}
		/**
		 * type and data_id
		 */
		$element['_render']['element'] .= sprintf( ' data-wpt-type="%s"', __FUNCTION__ );
		$element['_render']['element'] .= $this->_getDataWptId( $element );

		$element['_render']['element'] .= ' />';
		$pattern = $this->_getStatndardPatern( $element );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output . "\r\n";
	}

	/**
	 * Returns HTML formatted output for textarea element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function textarea( $element ) {
		$element['#type'] = 'textarea';
		if ( ! isset( $element['#attributes']['rows'] ) ) {
			$element['#attributes']['rows'] = 5;
		}
		if ( ! isset( $element['#attributes']['cols'] ) ) {
			$element['#attributes']['cols'] = 1;
		}
		$element = $this->_setRender( $element );
		$element['_render']['element'] = '<textarea id="' . $element['#id']
			. '" name="' . $element['#name'] . '"'
			. $element['_attributes_string'];
		/**
		 * type and data_id
		 */
		$element['_render']['element'] .= sprintf( ' data-wpt-type="%s"', __FUNCTION__ );
		$element['_render']['element'] .= $this->_getDataWptId( $element );

		$element['_render']['element'] .= '>';

		$element['_render']['element'] .= isset( $element['#value'] ) ? esc_attr( $element['#value'] ) : '';
		$element['_render']['element'] .= '</textarea>' . "\r\n";
		$pattern = $this->_getStatndardPatern( $element );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output . "\r\n";
	}

	/**
	 * Returns HTML formatted output for file upload element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function file( $element ) {
		$element['#type'] = 'file';
		$element = $this->_setRender( $element );
		$element['_render']['element'] = '<input type="file" id="'
			. $element['#id'] . '" name="' . $element['#name'] . '"'
			. $element['_attributes_string'];
		if ( isset( $element['#disable'] ) && $element['#disable'] ) {
			$element['_render']['element'] .= ' disabled="disabled"';
		}
		/**
		 * type and data_id
		 */
		$element['_render']['element'] .= sprintf( ' data-wpt-type="%s"', __FUNCTION__ );
		$element['_render']['element'] .= $this->_getDataWptId( $element );

		$element['_render']['element'] .= ' />';
		$pattern = $this->_getStatndardPatern( $element );
		$output = $this->_pattern( $pattern, $element );
		$output = $this->_wrapElement( $element, $output );

		return $output;
	}

	/**
	 * Returns HTML formatted output for markup element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function markup( $element ) {
		return $element['#markup'];
	}

	/**
	 * Returns HTML formatted output for hidden element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function hidden( $element ) {
		$element['#type'] = 'hidden';
		$element = $this->_setRender( $element );
		$output = '<input type="hidden" id="' . $element['#id'] . '" name="'
			. $element['#name'] . '" value="';
		$output .= isset( $element['#value'] ) ? $element['#value'] : 1;
		$output .= '"' . $element['_attributes_string'] . $this->_getDataWptId( $element ) . ' />';

		return $output;
	}

	/**
	 * Returns HTML formatted output for reset button element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function reset( $element ) {
		return $this->submit( $element, 'reset', 'Reset' );
	}

	/**
	 * Returns HTML formatted output for button element.
	 *
	 * @param array $element
	 *
	 * @return string
	 */
	public function button( $element ) {
		return $this->submit( $element, 'button', 'Button' );
	}

	/**
	 * Returns HTML formatted output for radio element.
	 *
	 * Used by reset and button.
	 *
	 * @param array $element
	 * @param string $type
	 * @param string $title
	 *
	 * @return string
	 */
	public function submit( $element, $type = 'submit', $title = 'Submit' ) {
		$element['#type'] = $type;
		$element = $this->_setRender( $element );
		$element['_render']['element'] = '<input type="' . $type . '" id="'
			. $element['#id'] . '" name="' . $element['#name'] . '" value="';
		$element['_render']['element'] .= isset( $element['#value'] ) ? $element['#value'] : $title;
		$element['_render']['element'] .= '"' . $element['_attributes_string']
			. ' />';
		$pattern = $this->_getStatndardPatern( $element, '<BEFORE><PREFIX><ELEMENT><SUFFIX><AFTER>' );
		$output = $this->_pattern( $pattern, $element );

		return $output;
	}

	/**
	 * Searches and returns submitted data for element.
	 *
	 * @param type $element
	 *
	 * @return type mixed
	 */
	public function getSubmittedData( $element ) {
		$name = $element['#name'];
		if ( strpos( $name, '[' ) === false ) {
			if ( $element['#type'] == 'file' ) {
				return $_FILES[ $name ]['tmp_name'];
			}

			return isset( $_REQUEST[ $name ] )
				? sanitize_text_field( $_REQUEST[ $name ] )
				: in_array( $element['#type'], array(
					'textfield',
					'textarea',
				) ) ? '' : 0;
		}

		$parts = explode( '[', $name );
		//https://icanlocalize.basecamphq.com/projects/7393061-toolset/todo_items/196173458/comments
		//Security Fixing
		//$parts = array_map(create function('&$a', 'return trim($a, \']\');'), $parts);
		$parts = array_map( "cred_mytrimfunction", $parts );
		if ( ! isset( $_REQUEST[ $parts[0] ] ) ) {
			return in_array( $element['#type'], array( 'textfield', 'textarea' ) ) ? '' : 0;
		}
		$search = $_REQUEST[ $parts[0] ];
		for ( $index = 0; $index < count( $parts ); $index ++ ) {
			$key = $parts[ $index ];
			// We're at the end but no data retrieved
			if ( ! isset( $parts[ $index + 1 ] ) ) {
				return in_array( $element['#type'], array( 'textfield', 'textarea' ) ) ? '' : 0;
			}
			$key_next = $parts[ $index + 1 ];
			if ( $index > 0 ) {
				if ( ! isset( $search[ $key ] ) ) {
					return in_array( $element['#type'], array( 'textfield', 'textarea' ) ) ? '' : 0;
				} else {
					$search = $search[ $key ];
				}
			}
			if ( is_array( $search ) && array_key_exists( $key_next, $search ) ) {
				if ( ! is_array( $search[ $key_next ] ) ) {
					return $search[ $key_next ];
				}
			}
		}

		return 0;
	}

	private function _getDataWptId( $element ) {
		$html = '';
		if ( array_key_exists( '#id', $element ) ) {
			if ( Toolset_Utils::is_real_admin() ) {
				$html .= sprintf( ' data-wpt-id="%s"', preg_replace( '/\[/', '-', preg_replace( '/\]/', '', $element['#name'] ) ) );
			} else {
				$html .= sprintf( ' data-wpt-id="%s_%s"', $this->_id, $element['#id'] );
			}
			if ( array_key_exists( '#name', $element ) && $element['#name'] ) {
				if ( ! Toolset_Utils::is_real_admin() && $this->_isRepetitive( $element ) ) {
					$html .= sprintf( ' data-wpt-name="%s"', preg_replace( '/\[.+$/', '', $element['#name'] ) );
				} else {
					if ( preg_match( '/^wpcf_post_relationship\[\d+\]\[\d+\]\[[^\]]+\]/', $element['#name'] ) ) {
						$html .= sprintf(
							' data-wpt-name="%s"', preg_replace( '/^wpcf_post_relationship\[\d+\]\[(\d+)\]\[wpcf-([^\]]+)\]/', "wpcf[$2-$1]", $element['#name'] ) );
					} else {
						$html .= sprintf( ' data-wpt-name="%s"', $element['#name'] );
					}
				}
			}
		}

		return $html;
	}

	private function _getStatndardPatern( $element, $default = false ) {
		if ( isset( $element['#pattern'] ) ) {
			return $element['#pattern'];
		}
		if ( $default ) {
			return $default;
		}
		if ( Toolset_Utils::is_real_admin() ) {
			return '<BEFORE><LABEL><DESCRIPTION><ERROR><PREFIX><ELEMENT><SUFFIX><AFTER>';
		}

		return '<BEFORE><DESCRIPTION><ERROR><PREFIX><ELEMENT><SUFFIX><AFTER>';
	}

	private function _isRepetitive( $element ) {
		if ( ! is_array( $element ) ) {
			return false;
		}

		return array_key_exists( '#repetitive', $element ) && $element['#repetitive'];
	}

	static function json_encode( $array ) {
		// php > 5.3 do not escape utf-8 characters using native constant argument
		if ( defined( 'JSON_UNESCAPED_UNICODE' ) ) {
			return json_encode( $array, JSON_UNESCAPED_UNICODE );
		} // fallback for php < 5.3 to support unicode characters in json string
		else {
			if ( function_exists( 'mb_decode_numericentity' ) ) {
				return self::json_encode_unescaped_unicode( $array );
			} else {
				return json_encode( $array );
			}
		}
	}

	/**
	 * @param $arr
	 *
	 * @return string
	 * courtesy from: http://www.php.net/manual/ru/function.json-encode.php#105789
	 */
	public static function json_encode_unescaped_unicode( $arr ) {

		array_walk_recursive( $arr, array( __CLASS__, 'json_unescaped_unicode_walk_callback' ) );

		return mb_decode_numericentity( json_encode( $arr ), array( 0x80, 0xffff, 0, 0xffff ), 'UTF-8' );
	}

	/*
	 * Helper function to json_encode with UTF-8 support for php < 5.3
	 */

	public static function json_unescaped_unicode_walk_callback( &$item, $key ) {
		if ( is_string( $item ) ) {
			$item = mb_encode_numericentity( $item, array( 0x80, 0xffff, 0, 0xffff ), 'UTF-8' );
		}
	}

}